理解java中【同步】和【死锁】
一.理解同步
要想解决资源共享的同步操作问题,可以使用两种方法:
- 使用同步代码块
之前学习过程中,代码块分为四种:
l 普通代码块:是直接定义在方法之中的;
l 构造块:是直接定义在类中的,优先于构造方法执行,会重复调用;
l 静态块:是使用static关键字声明的,优先于构造块执行,并且只执行一次;
l 同步代码块:是使用synchronized关键字声明的代码块,称为同步代码块
同步的时候必须指明同步的对象,一般情况下会将当前对象作为同步的对象,使用this关键字表示。 注意,同步会使程序运行变慢!
代码如下:
package cn.test.java.mutilthread;
class SyncThread2 implements Runnable{
private int ticket = 10;
public void run(){
for(int i = 0;i>10;i++){
synchronized (this) {
if(this.ticket>0){
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"剩下票数:"+ticket--);
}
}
}
}
}
public class SyncThreadDemo2 {
public static void main(String[] args) {
SyncThread2 t = new SyncThread2();
Thread t1 = new Thread(t);
Thread t2 = new Thread(t);
Thread t3 = new Thread(t);
t1.start();
t2.start();
t3.start();
}
}
- 使用同步方法
package cn.test.java.mutilthread;
class SyncThread3 implements Runnable{
private int ticket = 10;
public void run(){
for(int i = 0;i>10;i++){
this.sale();// 调用同步方法
}
}
public synchronized void sale(){ //声明同步方法
if(this.ticket>0){
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"剩下票数:"+ticket--);
}
}
}
public class SyncThreadDemo3 {
public static void main(String[] args) {
SyncThread3 t = new SyncThread3();
Thread t1 = new Thread(t);
Thread t2 = new Thread(t);
Thread t3 = new Thread(t);
t1.start();
t2.start();
t3.start();
}
}
二.理解死锁
死锁的出现是因为多个线程占用资源之后,没有进行释放,导致其他线程一直处于等待状态,在我们在代码中,出现死锁的原因有很多,极大多数是因为我们代码中编成的同步代码过多导致死锁的,
注意:使用synchronized同步代码中嵌入synchronized代码,非常容易导致死锁的出现。
代码如下:
class Zhangsan{ // 定义张三类
public void say(){
System.out.println("张三对李四说:“你给我画,我就把书给你。”") ;
}
public void get(){
System.out.println("张三得到画了。") ;
}
};
class Lisi{ // 定义李四类
public void say(){
System.out.println("李四对张三说:“你给我书,我就把画给你”") ;
}
public void get(){
System.out.println("李四得到书了。") ;
}
};
public class ThreadDeadLock implements Runnable{
private static Zhangsan zs = new Zhangsan() ; // 实例化static型对象
private static Lisi ls = new Lisi() ; // 实例化static型对象
private boolean flag = false ; // 声明标志位,判断那个先说话
public void run(){ // 覆写run()方法
if(flag){
synchronized(zs){ // 同步张三
zs.say() ;
try{
Thread.sleep(500) ;
}catch(InterruptedException e){
e.printStackTrace() ;
}
synchronized(ls){
zs.get() ;
}
}
}else{
synchronized(ls){
ls.say() ;
try{
Thread.sleep(500) ;
}catch(InterruptedException e){
e.printStackTrace() ;
}
synchronized(zs){
ls.get() ;
}
}
}
}
public static void main(String args[]){
ThreadDeadLock t1 = new ThreadDeadLock() ; // 控制张三
ThreadDeadLock t2 = new ThreadDeadLock() ; // 控制李四
t1.flag = true ;
t2.flag = false ;
Thread thA = new Thread(t1) ;
Thread thB = new Thread(t2) ;
thA.start() ;
thB.start() ;
}
};
三.总结
1.多个线程在访问同一资源的时候需要进行同步操作。
2.同步使用synchronized关键字完成,分为同步代码块及同步方法。
3.过多的同步有可能造成死锁的产生,死锁是在程序运行时的一种状态,了解就行了。
4.想停止线程,在代码中设置标志位flag,利用标志位来控制线程在生命周期。